--- tags: - operating systems - linux - storage - faq - repost --- [[toc]] ## 1 After searching a bit I could not find a simple and good howto to do that. The following method should work for any Linux distribution (Ubuntu, Debian, Manjaro, Archlinux, Fedora…). Source and target systems must be on the same processor architecture (though transfer from 32bit to 64bit should work). What you need: - 2 live USB keys (or cds) - To speed up data transfer: good quality ethernet cables (one cable between the 2 computers is OK), or a usb key/drive with a BIG ext4 partition. You can try over wifi, but it may be slow. ## 1. Boot source and target machines on live USB/CD Any live USB/CD should be OK. On the target computer, you will need a tool to partition your hard drive, like [gparted](http://gparted.org/ "gparted"). [rsync](http://rsync.samba.org/ "rsync") is also required for data transfer: it’s included in many live systems. [Ubuntu](http://www.ubuntu.com/ "Ubuntu") live cd is OK, [Manjaro](http://manjaro.org/ "Manjaro") live cd too. ## 2. Partition your target hard drive Use a tool like [gparted](http://gparted.org/ "gparted") to partition the target hard drive, with the same partitions as your source system (slash, swap, home…). I recommend you to assign LABELs to your partitions: for the fstab, it’s easier than UUIDs. ## 3. Mount all partitions on both machines On both systems, open a root terminal. Then, for each data partition (you can ignore `swap`): ```bash $ mkdir /mnt/slash $ mount /dev/sdaX /mnt/slash ``` If you have a home partition: ```bash $ mkdir /mnt/home $ mount /dev/sdaY /mnt/home ``` ## 4. Transfer the data (network or usb) This part may be tricky. Choose the method you prefer. ### Network 1. **Setup the network.** Test the connectivity with ping command. The easier is to plug the PCs on a DHCP network (like your ISP box) so that you get automatic IP addresses. If you linked the 2 pcs with a single cable, you’ll have to setup the IPs with NetworkManager (static ips, or adhoc network). 2. On source system, as root, create a simple `/etc/rsyncd.conf` file: ```toml uid = root gid = root use chroot = no [all] path = / ``` 3. Then start the rsync daemon server: `rsync --daemon` 4. On target PC, for each partition: `rsync -avHX SOURCE_IP::all/mnt/slash/ /mnt/slash/` Don’t forget `/` at the end of paths. `-a` will preserve many file attributes like owner and permissions, `-H` will preserve hardlinks if any, `-X` will preserve extended attributes like setuid. You may also add `-A` if you are using acls. What is good with rsync is that you can stop and restart the transfer whenever you want. ### USB Prepare a USB drive with a BIG `ext4` partition. 1. Mount the USB partition on source system (`mount /dev/sdbX /mnt/usb`) 2. For each partition: `rsync -avHX /mnt/slash/ /mnt/usb/slash/` 3. umount, unplug and remount the USB disk on the target system. 4. For each partition: `rsync -avHX /mnt/usb/slash/ /mnt/slash/` ## 5. Change fstab on target system As root, edit `/mnt/slash/etc/fstab` For each partition (including swap), replace the first field with the new UUID or LABEL (it’s straightforward with LABELs): `UUID=the-long-uuid`, or `LABEL=yourlabel` 2 ways to get the UUIDs / LABELs: ```bash $ ls -l /dev/disk/by-uuid/ $ blkid /dev/sdaX ``` ## 6. Reinstall Grub We will use a **chroot** (changed root environment) to be able to call the [grub](http://www.gnu.org/software/grub/ "Grub") install inside the migrated system. First, bind mount some system directories needed by grub, then chroot: ```bash $ mount --bind /proc /mnt/slash/proc $ mount --bind /sys /mnt/slash/sys $ mount --bind /dev /mnt/slash/dev $ mount --bind /run /mnt/slash/run $ chroot /mnt/slash ``` Then install grub in the [Master Boot Record](http://en.wikipedia.org/wiki/Master_boot_record "master boot record") of your hard drive, and update grub config file (with the new uuids…): ```bash $ grub-install /dev/sda $ update-grub ``` ## 7. Reboot target machine That’s it! Your system should be working on the new computer now. ## 2 Use clonezilla ## 3 This can be the fastest way to move. As to copy your hard drive partitions as disk images are quite fast. If you don't want to re-install every piece of software. Though creating, resizing and moving the disk images can take quite a long time. I would only recommend this if you are not going to upgrade to a new version of Ubuntu. Make sure you understand disk partitions and grub. Most of what I am doing will use the command line. You need to make sure you understand what a command does before you run it. I am not responsible for data loss as a result of the instructions that follow. Step one create a disk image of your installation. Fist we need to get some information about the setup. Using `parted -l` and `mount` ```bash $ sudo parted -l Model: ATA ST9320423AS (scsi) Disk /dev/sda: 320GB Sector size (logical/physical): 512B/512B Partition Table: msdos Number Start End Size Type File system Flags 1 32.3kB 197MB 197MB primary ext4 boot 2 197MB 10.2GB 10.0GB primary linux-swap(v1) 3 10.2GB 50.2GB 40.0GB primary ext4 4 50.2GB 299GB 249GB extended 5 50.2GB 54.4GB 4195MB logical ext4 6 54.4GB 65.9GB 11.5GB logical ext4 7 65.9GB 299GB 233GB logical ext4 $ mount /dev/sda5 on / type ext4 (rw,errors=remount-ro) /dev/sda7 on /home type ext4 (rw) /dev/sda1 on /boot type ext4 (rw) /dev/sda6 on /usr type ext4 (rw) # I took out the entries that were not need for these instructions $ cat /etc/fstab proc /proc proc nodev,noexec,nosuid 0 0 UUID=ddc8c237-e8ac-4038-a0ed-f7c866d6603b / ext4 errors=remount-ro 0 1 UUID=aa9881d1-5cc1-4e94-8cd7-8125e18ece2f /boot ext4 defaults 0 2 UUID=31a6fde1-6b96-4cc3-acfd-88573f52be36 /home ext4 defaults 0 2 UUID=073146a7-5668-4728-9a6f-1a599f358a8d /usr ext4 defaults 0 2 UUID=540b96b6-b3c3-4092-b4ad-6b33bcbbe16d none swap sw 0 0 ``` Your set up might look different. I have a separate partition for `/home`, root (`/`), and `/usr`. ## Creating the Disk Images I use `dd` as it is simple and quick. Make sure you read and understand how it works. You will need an empty partition that is bigger than the entire partition size that you are copying. This can take quite some time. Creating resizing and copying the partitions can take a few hours depending on their size. You will need to replace the external drive with a part to the storage media you will use for this process. ```bash sudo dd if=/dev/sda5 of=/media/externaldrive/sda5-root.img sudo dd if=/dev/sda7 of=/media/externaldrive/sda7-home.img sudo dd if=/dev/sda6 of=/media/externaldrive/sda6-usr.img ``` Here is an actual example of out put after running this on my set up. ```bash $ sudo dd if=/dev/sda5 of=/media/home0/sda5-root.img 8193087+0 records in 8193087+0 records out 4194860544 bytes (4.2 GB) copied, 55.3159 s, 75.8 MB/s ``` We can reduce the size of this disk image, using the tools provided by Linux. ``` $ sudo resize2fs -P sda5-root.img resize2fs 1.41.11 (14-Mar-2010) Estimated minimum size of the filesystem: 605972 $ ls -sh ./sda5-root.img 4.0G ./sda5-root.img $ sudo resize2fs -M sda5-root.img resize2fs 1.41.11 (14-Mar-2010) Please run 'e2fsck -f sda5-root.img' first. $ sudo e2fsck -fy ./sda5-root.img # y makes it run without asking thousands of questions. ``` `e2fsck` will output lots of errors or fixes necessarily. This is because the information in the file system is no longer correct in terms of where the partition boundaries start and end. This is correct because it is no longer in the partition it was configured for. ``` $ sudo resize2fs -M sda5-root.img resize2fs 1.41.11 (14-Mar-2010) Resizing the filesystem on sda5-root.img to 605505 (4k) blocks. Resizing the filesystem on sda5-root.img to 605505 (4k) blocks. The filesystem on sda5-root.img is now 605505 blocks long. $ ls -sh ./sda5-root.img 2.4G ./sda5-root.img ``` It essentially removes all the free space in the partition. So for the larger partition, this can be more the 50% of the disk size. Much quicker to copy a smaller file You now need to boot up your new laptop with a live disk and do what follows here. You need to use a live disk as you can not make changes to a running partition that is currently used by the installed operating system. You can now copy these disk images into the partitions on the new computer. You should have set up these partitions already. Using the live disk and `gparted` is a quick and easy way to do this. Make sure you have all the partitions your system requires. You can make these partitions larger than the ones you had on your previous system. When we copy the disk images into them, we will resize the file system and it will take up all the free space on the partition. Now step two: copying the disk images on to the new drive and into the new partitions. ``` sudo dd if=/media/exteranldrive/sda5-root.img of=/dev/sda3 # replace the [sda3] with your partition. ``` On my machine, this is what the output looked like ``` $ sudo dd if=./sda5-root.img of=/dev/sdb6 4844040+0 records in 4844040+0 records out 2480148480 bytes (2.5 GB) copied, 87.4921 s, 28.3 MB/s $ sudo fsck.ext4 -fy /dev/sdb6 e2fsck 1.41.11 (14-Mar-2010) Pass 1: Checking inodes, blocks, and sizes Pass 2: Checking directory structure Pass 3: Checking directory connectivity Pass 4: Checking reference counts Pass 5: Checking group summary information root1: 50470/504000 files (1.4% non-contiguous), 616736/2060328 blocks ``` Now we need to edit the fstab file to point to the correct devices. If you have just copied the new disk partition on to your new disk, the fstab file is on that partition so you need to mount it in order to access the file. You will also need to have the root partition mounted in order to install grub if you don't have a separate boot partition. ``` $ sudo mkdir /mnt/tmp mount /dev/sdb6 /mnt/tmp $ sudo blkid # to see what the disk uuid is /dev/sda5: LABEL="root1" UUID="ddc8c237-e8ac-4038-a0ed-f7c866d6603b" TYPE="ext4" /dev/sdb6: LABEL="root1" UUID="ddc8c237-e8ac-4038-a0ed-f7c866d6603b" TYPE="ext4" $ gksu gedit /mnt/tmp/etc/fstab replace the UUID with the UUID of your partition UUID=ddc8c237-e8ac-4038-a0ed-f7c866d6603b / ext4 errors=remount-ro 0 1 ``` Here you can see that the new disk image that I copied across to the other disk has the same UUID as the the original file system. So you could copy your fstab file form your old install across into your new install and have a working system. That will boot. On my set up I can't leave my computer like this or it will boot to whichever device it finds first. Edit fstab and make sure the uuid match the partitions that you have set up for root and home and whatever other partition you set up. Last step is to install grub on you new disk. ``` sudo chroot /mnt/tmp # your root partition. grub-install /dev/XXX ``` In my case: ``` grub-install /dev/sdb update-grub ``` Please read these instructions before beginning. It is no use having all the data on your new laptop and not being able to boot it up. [https://help.ubuntu.com/community/Grub2](https://help.ubuntu.com/community/Grub2) [https://help.ubuntu.com/community/RecoveringUbuntuAfterInstallingWindows](https://help.ubuntu.com/community/RecoveringUbuntuAfterInstallingWindows) ## 4 Moving a Linux installation from one machine to another is actually relatively easy to do, but there aren’t many articles online that walk through the whole process. Unlike some other operating systems (I’m looking at you Windows) Linux is by default fairly uncoupled from the hardware it is running on. That said, there are still a few gotchas that need to be watched out for, especially when it comes time to configure the bootloader. This post takes you through the whole process and assumes minimal Linux experience: if you’re comfortable with basic shell commands you should be able to follow along. Since there are a lot of different reasons to want to clone a system we’ll be focusing on actually understanding what each step is doing so that you can adapt what I’ve described to your situation. While I’m assuming you’re using physical machines here, this procedure works just as well with VMs, whether run locally via something like VirtualBox or VMs provided by a cloud provider like Amazon AWS. If you find yourself needing to move from one cloud provider to another, you can adapt the steps in this guide to make that happen, just keep in mind that on a cloud VM it may be difficult to boot into a livecd so you will probably need to instead attach two hard drives to the VM–one with a fresh Ubuntu install that can act as your “livecd” and an empty one will be used as the restore target. I’ve listed out the commands to clone a system with minimal explanation as a reference below. If you know your way around Linux you may be able to just run through these commands, adapting as needed to fit your situation. If you’d like more detail, keep reading and we’ll go over exactly what each command is doing (and why it’s needed) below. > 1. Bind-mount the source drive to a new location so that we don’t end up in an infinite copy loop while copying `/dev/zero`, etc.: > > mount --bind / /mnt/src > > 2. `tar` up the source filesystem: > > tar -C /mnt/src -c . > source-fs.tar > > and copy the resulting `source-fs.tar` onto a USB drive or network share that you can access from the destination machine. > > 3. On the dest machine boot from a live-cd (I used the Ubuntu install disc) > > 4. Partition the drive on the destination machine. The easiest way to do this is to use `gparted` (included on the Ubuntu live-cd). How you partition will differ depending on whether you want to use MBR or EFI mode: > > - **MBR mode**: just create one big ext4 partition on your target drive, and use use `gparted`’s ‘Manage Flags’ right click menu to add the `boot` flag > - **EFI mode**: create one 200-500MB vfat/fat32 partition (use `gparted`’s ‘Manage Flags’ right click menu to add `boot` and `esp` flags), and create one ext4 partition in the remaining space. > 5. Once booted into the live-cd, mount your destination filesystem. I’m mounting mine at `~/dest`. > > mount /dev/ ~/dest > > 6. Use `tar` to extract your image onto the destination filesystem, (using `pv` to provide a progress meter since this can take a while): > > pv < [image-file] | tar -C ~/dest -x > > 7. `chroot` into the newly extracted filesystem > > cd ~/dest > for i in /dev /dev/pts /proc /sys /run; do sudo mount --bind $i .$i; done > mkdir -p ./boot/efi # skip if using MBR mode > sudo mount /dev/ ./boot/efi # skip if using MBR mode > sudo chroot . > > 8. Run `grub-install` from inside the chroot: > > apt install grub-efi-amd64-bin # skip if using MBR mode > grub-install /dev/ # use the whole drive (e.g. sda, not sda1) > # Step 1: Bind mount the root filesystem The first command we run is `mount --bind / /mnt/src`. In Linux-land filesystems are accessed by mounting them to a path (usually under `/media` or `/mnt`). Here we’re using something called a bind mount, which allows you to “bind” a mount point to another mount point. In other words, you can access the same folder at two locations. In this instance, we are telling the system to make the `/` folder available at `/mnt/src` as well. If you write a file to `/test-file`, you’ll see that it’s also available at `/mnt/src/test-file`. Why is this needed you ask? Well, when a Linux system boots it creates some virtual filesystems that many Linux programs rely on. One of the more commonly used ones is the `/dev` folder, which is how Linux addresses the physical hardware installed in your system. The files in the `/dev` folder aren’t real files though, so it doesn’t make sense to copy them to another system–that system will have it’s own `/dev` that reflects it’s own hardware. More importantly for our current purposes, `/dev` also contains some special “files” such as `/dev/zero`, which returns an infinite amount of zeros, and it’ll take more time than any of us have to copy an infinite amount of zeros. Bind mounting `/` to `/mnt/src` allows us to sidestep this issue: this system’s `/dev` will still exist at `/dev`, but you won’t find a corresponding `/mnt/src/dev/zero` folder, so copying from `/mnt/src` avoids starting an infinitely long copy process. # Step 2: `tar` up the source file system Now that we’ve got the filesystem bind-mounted we can start preparing our image. All we really need to do here is save the contents of the root filesystem (excluding special filesystems such as `/dev`) into a tar archive: ```bash tar -C /mnt/src -c . > source-fs.tar ``` The `-C` flag tells `tar` to change directories to `/mnt/src`, `-c` tells tar to use ‘create’ mode (as in, create a tar archive, not extract one) and the `.` tells it to do so in the current directory (which is now `/mnt/src` thanks to our `-C` flag). We then use shell redirection via the `>` sign to write the output to the file `source-fs.tar`. **Make sure `source-fs.tar` is not on the same drive you are copying from** or you may kick off another infinite loop! > **NOTE:** In this example I’m just writing the image to a file, but if you wanted you could also stream the filesystem directly to another machine over the network. The most common way to to this is to use `ssh` and a shell pipe like so: > > tar -C /mnt/src -c . | \ > ssh 'tar -C -x' > > This uses a shell pipe to send the output of `tar` into the ssh command, which takes care of setting up an encrypted connection to the other machine, and then runs `tar -C -x` on the other machine, connecting the stdin of `tar` on the remote machine to the stdout of `tar` on the sending machine. ### Step 3: On the dest machine boot from a live-cd On the destination machine (the machine we want to clone our system _to_), we need to boot into an operating system that is not running off of the system’s primary hard drive, so that we can write our cloned image into the new drive. I usually just grab the latest Ubuntu live-cd from [Ubuntu’s website](https://ubuntu.com/download/desktop) website and write it to a USB via [Etcher](https://www.balena.io/etcher/) or the `dd` command. Ubuntu provides directions on how to prepare an Ubuntu LiveUSB [here](https://tutorials.ubuntu.com/tutorial/tutorial-create-a-usb-stick-on-ubuntu). If you don’t like Ubuntu any Linux livecd should work fine, just make sure it has a partitioning tool like `gparted` (gui) or `fdisk` (cli). ### Step 4: Partition the drive on the destination machine Here is where things start to get a little tricker. **There are two common ways to boot a Linux system, MBR (an older method) or EFI (a newer method), and each have different partitioning requirements.** If possible you’ll want to use EFI, but if you have an older machine that doesn’t support EFI mode you may need to use MBR. The easiest way to check if a machine supports EFI mode is to boot into the Ubuntu livecd and check if a directory called `/sys/firmware/efi` exists: $ ls /sys/firmware acpi devicetree dmi efi memmap If there’s no `efi` folder in `/sys/firmware` then you’re on an MBR machine. If there is an `efi` folder present, then you’re on an EFI machine and we’ll need to create an EFI partition as well as a root partition. From the Ubuntu livecd open a terminal and let’s fire up [gparted](http://todo/) on the drive we’re going to partition: sudo gparted Using the selector in the upper left, choose the drive you’re going to be restoring to. On my system this is `/dev/nvme0n1`, but depending on the hardware in you’re machine you may have a different designation such as `/dev/sda`. Once you have your drive selected, choose `Device -> Create Partition Table` from the Device menu. You’ll be greeted with a scary looking screen like the following: ![]() Make sure you have the right drive selected here, because, as the window above indicates, as soon as you hit apply `gparted` will proceed to erase everything on that drive. Because the MBR approach is how MS-DOS historically loaded itself, some tools (including gparted) refer to MBR partition layouts as `msdos`. If your system is an MBR system, then leave that unchanged, otherwise select `gpt` from the list since GPT is the hard-drive layout that works with EFI. For the rest of this step, we will proceed with an EFI based install. If you’re doing an MBR install then you can skip the create EFI partition portion. In the next screen we’ll need two create two partitions, one ~500MB EFI partition (this can be smaller if you need to save space, but things may break if you make it less than 200MB) and a second partition filling up the remainder of the drive. This second partition is the partition we will restore our clone into. Let’s start by creating the EFI partition. Use the menus to choose `Partition -> New`, and in the screen that follows set the size to 500MB, and set the file system to `fat32` which is the filesystem type EFI requires. Repeat the process for the second partition, but this time do not enter a size and choose ext4 for the filesystem type. When you’re finished your partition layout should look similar to the below: ![]() Go ahead and use the `Edit -> Apply all Operations` menu to write the new partition table. Once that’s completed we have to set some partition flags to make the drives properly bootable. To do this, right click on the first fat32 partition and choose ‘Manage Flags’. Click the checkmark next to `boot` (which may also automatically check the ‘esp’ flag) and hit Close. Keep track of the device names (they will show in the Partition column with names that start with `/dev/`) as you will need them for the next step. # Step 5: Mount the destination filesystem At this point our target system is prepared and we are ready to restore the image onto this machine. Before we can do anything with the new hard drive layout we need to mount it. Boot back into the Ubuntu livecd if you’re not already in it, and open up a terminal window. We’ll first create a mount point (an empty directory) where we’ll mount the two drives. I’m using `~/efi` and `~/dest` mkdir ~/efi mkdir ~/dest And then mount the drives to them. On my system the drive I was partitioning was `/dev/sdb`, so my EFI and data partitions are `/dev/sdb1` and `/dev/sdb2` respectively. Your system may assign different identifiers, make sure to use the names shown by `gparted` in the `Partition` column: mount /dev/sdb1 ~/efi mount /dev/sdb2 ~/dest # Step 6: Use `tar` to extract your image to the destination filesystem Now that we have all our mount points set up, we can do the reverse of the image creation process from step 2 to duplicate our source machine’s filesystem onto the new machine. Since this can take a while I like to use a tool called `pv` (pv stands for pipe viewer) to provide a progress meter. You can install `pv` by doing `sudo apt update && sudo apt install pv`. Once `pv` is installed, we can start the restore process. First, find a way to get the Ubuntu livecd access to the source image we created in Step 2. Most likely this means plugging a USB drive into the machine. Once you have access to the image file run the following command, replacing `[image-file]` with the path to your source tar file: pv < [image-file] | tar -C ~/dest -x The above command is saying to take the contents of `[image-file]` and send it to `pv` over stdin. `pv` reads the data from the file, prints out a nice progress meter, and then sends the data it’s reading to `tar` via a shell pipe (the `|` symbol). `-C` then tells `tar` to first change directories to `~/dest` (where we mounted our destination partition in the previous step), and the `-x` tells `tar` to run in extract mode. This may take a while, but when the process completes you will have completely restored all the files that originally lived on the source machine onto the new machine. Getting the files there is only half the battle however, we still need to tell Linux how to boot into this filesystem, which we’ll do in the next step. # Step 7: `chroot` into the newly extracted filesystem to install a bootloader At this point we have all the files we need on the new system, but we need to make the new system bootable. The easiest way to do this is to piggyback off of the Ubuntu livecd’s kernel, and use the `chroot` command to make our current Linux installation (the Ubuntu livecd) pretend like it’s the installation we just copied over to the new machine. For this to work we have to use our helpful friend `mount --bind` again to do the reverse of what we did in step 1. This time rather than avoiding copying these special filesystems, we instead want to give the `chroot`-ed installation temporary access to the special filesystems that our Ubuntu livecds created so that it can act as a functional Linux installation. First, change directories to where the new installation is mounted (`~/dest` if you followed the example above): cd ~/dest Then we’ll use `mount ---bind` to give the chroot access to the linux special directories: for i in /dev /dev/pts /proc /sys /run; do sudo mount --bind $i .$i; done > **NOTE:** We use a `for` loop here to save ourselves some typing, but the above line is just telling the system to run the command `sudo mount --bind ./` for each of the special directories listed between the `in` and the `;` . In other words, the single line above is equivalent to running the following: > > sudo mount --bind /dev ./dev > sudo mount --bind /dev/pts ./dev/pts > sudo mount --bind /proc ./proc > sudo mount --bind /sys ./sys > sudo mount --bind /run ./run If installing in EFI mode we also need to give our chroot access to the EFI partition we mounted earlier. `mount --bind` comes to the rescue again here, we simply bind mount the livecd mount point into the `/boot/efi` directory inside the chroot (`/boot/efi` is where grub expects to find the EFI partition). cd ~/dest mkdir -p boot/efi mount --bind ~/efi boot/efi Now that we have access to the Linux special folders (and the EFI partition), we can use the `chroot` command to actually use our source installation: sudo chroot ~/dest At this point you should have a shell inside the same Linux environment you originally copied. Try running some programs or looking at some files that came from your old machine. GUI programs may not work properly, but other then that you should have a fully functional copy of your old installation. Booting into an Ubuntu livecd and running the above `chroot` commands every time you want to use this machine is not very practical though, so in the next step we’ll install the grub bootloader to make it into a full-fledged bootable Linux installation. # Step 8: Run `grub-install` from inside the chroot Grub is the most common Linux bootloader and is what we’ll use here. Grub has an MBR flavor and an EFI flavor. If the machine you cloned from was running Ubuntu it most likely already has grub installed, but may not have the EFI version of grub installed. Run the following to install the EFI version (feel free to skip if you’re doing an MBR clone): apt install grub-efi-amd64-bin # skip if using MBR mode If your source distro is not Ubuntu based make sure to fully install grub via your distro’s package manager first. Once you have grub fully installed then you just need to run `grub-install` against the drive you installed to. In my case that’s `/dev/sdb`, but this may be different on your machine. If unsure fire up `gparted` as we did in Step 4 and check the names listed in the partition column there. Next we install grub to our drive, thereby making it bootable. Be careful to install grub to a **drive** and not to a partition. Partitions will usually have a number on the end while a drive will usually end with a letter (e.g. `/dev/sdb`, not `/dev/sdb1`). grub-install /dev/sdb update-grub If all went well you will see messages saying grub was successfully installed. When you see this feel free to reboot and check out your freshly cloned installation. If you got error messages and are installing in EFI mode it’s possible grub tried to use MBR mode. It might be worth a try running `grub-install` this way to force EFI mode: grub-install --target=x86_64-efi # Wrapping up That’s it, at this point you should have a fully operational clone of your original system, and hopefully also have a solid understanding of each step in the clone process and why it’s needed. Once you realize that a Linux installation is really just a filesystem and a mechanism for booting it, tools like Docker start to make a bit more sense: a docker image is basically just fancy version of the `tar` image we created here, with some changes to handle docker layers and make the image files easier to distribute. In fact, just as we were able to “run” the system we installed via `chroot` before we actually made it bootable, you can convert the tar image we created into a docker container quite easily: docker import [image-file] 99% of the time you’re better off just using a `Dockerfile` and docker’s own tooling to build your images, but if you need a quick and dirty way to “dockerize” an existing server you could do this without even having to shut down the existing server! Similarly, the `docker export` command can export a tarball like the one we created for any docker image. Once you extract it you could use the same `mount --bind` and `chroot` dance we did above to get a shell inside the “container.” If you wanted to get a bit crazy, you could even use the steps from this guide to restore a tarball exported from a docker image onto a physical machine and run it on bare metal. In real life this won’t work with many/most docker images because (for space conservation reasons) many docker images strip out some of the files needed to support physical booting, so you may be asking for trouble if you try this in real life.